JavaScript CodeStyle
##Overview
###Scope
This document provides the coding standards and guidelines for developers and teams working for or with VG. The subjects covered are:
- Javascript file formatting
- Naming conventions
- Coding style
- Inline documentation
- Errors and exceptions
###Goals
Good coding standards are important in any development project, particularly when multiple developers are working on the same project. Having coding standards helps to ensure that the code is of high quality, has fewer bugs, and is easily maintained.
##General
- Files should be encoded in UTF-8 without BOM.
- The line-break character must be LF -
\n
. - Files should end with a LF character.
- One level of indentation is achieved with 4 space characters.
- Lines should be no longer than 120 characters.
- Trailing whitespace at the end of lines should be removed.
##Naming
- All files, functions, variables, parameters, classes etc should be in English, and use generic terms. Exceptions are allowed where the name is intended to reflect very specific Norwegian names or concepts, e.g. "skattelister", "rampelys" etc.
variableNamesLikeThis
functionNamesLikeThis
ClassNamesLikeThis
methodNamesLikeThis
CONSTANTS_LIKE_THIS
namespacesLikeThis
events-like-this
private
or protected
properties and methods should be prefixed with a single _
character- Shortened and abbreviated names should be avoided.
- Common abbreviations, such as
JSON
and XML
are written in CamelCase
. For example: Json
, Xml
. - Filenames should not be camelCased and spaces are prohibited. Use the
-
character instead. - Most functions should be named. This is especially important when working with asyncronous processes like timers and callbacks. This helps greatly when debugging:
Bad:
setTimeout(function() {
doThatThing();
}, 1000);
Good:
setTimeout(function someDescriptiveTitle() {
doThatThing();
}, 1000);
##Strictness
Always start the top level block with 'use strict;'
. If writing CommonJS modules, assume a scope is present and place the pragma at the top of the file. If writing AMD or standalone Javascript, ensure your file has a separate scope and apply the pragma as early as possible:
CommonJS:
'use strict';
var fs = require('fs');
AMD/standalone:
(function() {
'use strict';
var foo = 'bar';
});
##Variable declaration
Good:
var keys = ['foo', 'bar'];
var values = [23, 42];
var object = {};
while (items.length) {
var key = keys.pop();
object[key] = values.pop();
}
Bad:
var keys = ['foo', 'bar'],
values = [23, 42],
object = {},
key;
while (items.length) {
var key = keys.pop();
object[key] = values.pop();
}
##Literals
###Objects
- There should be no whitespace characters before the colon:
var obj = {
prop: 0
};
- Only property names should be aligned within object literals:
Good:
var obj = {
a: 0,
b: 1,
lengthyName: 2
};
Bad:
var obj = {
a : 0,
b : 1,
lengthyName: 2
};
- Quotes around property names should be typed only if needed:
Good:
var obj = {
key: 0,
'key-key': 1
};
Bad:
var obj = {
'key': 0,
'key-key': 1
};
- There should be no trailing commas after the last value of the object, as this breaks older IEs and certain Javascript parsers:
Bad:
var myObject = {
key: 'value',
key2: 'value',
};
- Reserved words must be quoted if using them as properties:
Good:
var obj = {
'default': 'someValue',
'super': 'someSuper'
};
Bad:
var obj = {
default: 'someValue',
super: 'someSuper'
};
###Arrays
- When enumerating elements in an array literal, spaces should be typed after the comma only:
var fellowship = ['foo', 'bar', 'baz'];
###Strings
- String literals should use single quotes:
var lyrics = 'Never gonna give you up. Never gonna let you down. Never gonna turn around and desert you.';
- If a string contains a single quote character, it should be escaped:
var test = 'It shouldn\'t fail';
##Semicolons
Statements should always end with a semicolon.
##Keywords
- Keywords are always followed by a single space character, except after anonymous functions:
if (test) {
}
function foo() {
}
var bar = function() {
};
- If the keyword is followed by a semicolon, there should be no space between them:
return;
##Block Statements
- The opening curly brace should be on the same line and separated with one space character:
if (test) {
}
function foo() {
}
- Branching and looping statements should always be surrounded with curly braces:
Good:
if (test) {
return;
}
Bad:
if (test)
return;
if (test) return;
if (test) { return; }
##Conditional Statements
###if
- The
else
keyword should be on the same line as the closing brace of the if-part of the statement:
if (test) {
} else {
}
- Condition statements should not contain assignment operations:
Good:
var foo = bar();
if (foo > 0) {
}
Bad:
var foo;
if ((foo = bar()) > 0) {
}
- Logical operators should not be used for conditional branching:
Good:
if (condition) {
actionIfTrue();
} else {
actionIfFalse();
}
Bad:
condition && actionIfTrue() || actionIfFalse();
if (longCondition ||
anotherLongCondition &&
yetAnotherLongCondition
) {
}
Good:
if (getType() === 'driving') {
}
Bad:
if ('driving' === getType()) {
}
###switch
The switch statement should be written as in the example:
switch (value) {
case 1:
break;
case 2:
break;
default:
}
##Loops
###for
Functional constructs (map, reduce etc) are prefered where applicable. Array.prototype.forEach is prefered over the for
loop.
[1, 2, 3].forEach(function (value) {
console.log(value);
});
Performance-critical parts of the code can use a for
statement, as well as cases where the loop can be broken out of before iteration has completed.
###for (var i in obj)
When using the for-in
loop, authors should be aware of the pitfalls of enumerable properties on prototypes and know when to use hasOwnProperty().
##Operators and keywords
###'const' keyword
Never use the const keyword.
###'with' operator
The with
operator should not be used.
###Comparison Operators
If there is no need for type casting, the strict equality operator ===
(or strict inequality !==
) should be used.
###Ternary Operator
The ternary operator should be written as in the examples:
var x = a ? b : c;
var y = a ?
longButSimpleOperandB : longButSimpleOperandC;
var z = a ?
moreComplicatedB :
moreComplicatedC;
###Unary Operators
Unary operators should be typed without whitespace between them and their operands:
var foo = !bar;
Exceptions from this rule are the unary special JS operators).
##eval
The eval
function should be avoided.
json
serialized data should be parsed with JSON.parse.
##undefined
Checking for undefined
values can be done using either the strict equality operator or using typeof
.
Good:
x === undefined;
Accepted:
typeof x === 'undefined'
Bad:
x === void 0
##Parentheses
- Should be used only if it is required of the expression's syntax or semantics.
- Should not be used with the unary operators
delete
, typeof
and void
, or with the keywords return
, throw
and new
.
##Exceptions
throw
should be used with new Error
or an object of a class derived from Error
:
Good:
throw new Error('msg');
Bad:
throw 'msg';
##Type Casting
Type casting should be done explicitly (note the difference between parsing and type casting):
Good:
Boolean(foo)
Number(bar)
String(baz)
[].indexOf(qux) === -1 or [].indexOf(qux) < 0
Bad:
!!foo
+bar
baz + ''
~[].indexOf(qux)
##Parsing
- When parsing strings to integers or floats, a radix MUST be specified.
Good:
var width = parseInt('20px', 10);
var someNum = parseInt('0x10', 16);
Bad:
var width = parseInt('20px');
var someNum = parseInt('0x10');
##Multi-Line Statements
- If a statement is longer than the maximum line length, it is split into several lines and properly indented.
- Lines of the statement should be split after an operator:
var debt = this.calculateBaseDebt() + this.calculateSharedDebt() + this.calculateDebtPayments() +
this.calculateDebtFine();
- Closing parentheses should be on a new line with the indentation of the current block statement:
Good:
DoSomethingThatRequiresALongFunctionName(
very_long_argument1,
argument2,
argument3,
argument4
);
anotherStatement;
Bad:
DoSomethingThatRequiresALongFunctionName(
very_long_argument1,
argument2,
argument3,
argument4);
anotherStatement;
##Method Chaining
When a method is called on a new line, it should:
- Be one indentation level deeper than the target object.
- Begin with the property access operator
.
.
Good:
someObject
.operation()
.operationWithCallback(function (obj) {
obj.processed = true;
})
.end();
Bad:
someObject.
start().
end();
someObject
.start()
.end();
##String concatenation
- Strings should be concatenated with the
+
operator. - The
[].join('')
should be avoided. - Escaping newline literals inside strings should be avoided.
Good:
var foo = 'A rather long string of English text, an error message ' +
'actually that just keeps going and going -- an error ' +
'message to make the Energizer bunny blush (right through ' +
'those Schwarzenegger shades)! Where was I? Oh yes, ' +
'you\'ve got an error and all the extraneous whitespace is ' +
'just gravy. Have a nice day.';
Bad:
var foo = 'A rather long string of English text, an error message \
actually that just keeps going and going -- an error \
message to make the Energizer bunny blush (right through \
those Schwarzenegger shades)! Where was I? Oh yes, \
you\'ve got an error and all the extraneous whitespace is \
just gravy. Have a nice day.';
##Empty Lines
A single empty line can be used as a separator for grouping the code into logical blocks:
doSomethingTo(x);
doSomethingElseTo(x);
andThen(x);
nowDoSomethingWith(y);
andNowWith(z);
##Function Context
doAsync(function () {
this.fn();
}.bind(this));
- Preferably, the context argument should be used (if available):
Good:
[1, 2, 3].forEach(function (n) {
this.fn(n);
}, this);
Bad:
[1, 2, 3].forEach(function (n) {
this.fn(n);
}.bind(this));
- If assigning the current context to a variable (generally avoided), the variable should be named
_this
:
var _this = this;
doAsync(function () {
_this.fn();
});
##Comments
- In-line comments should start with
//
. Between the //
and the text of the comment should be one space character. - Comments for functions, classes, etc. should be written according to the jsdoc documentation syntax.
##Classes
- "Symmetrical" methods should be declared one after the other. For example:
var FooClass = inherit({
__constructor: function () {},
destruct: function () {},
someMethod: function () {}
});
##node.js
###Importing Modules
- Modules should be imported in the beginning of the file, after the description of the module (if present):
Good:
var http = require('http');
var fs = require('fs');
Bad:
var http = require('http');
var fs = require('fs');
This rule does not apply to modules that are imported "on demand".
- Module import calls should be grouped according to the following order:
- Standard node.js modules (i.e. fs, util, etc.).
- External lib modules.
- Modules of the current application.
Linting
We encourage and suggest our developers to use ESLint as the primary linting tool, since it supports the largest number of rule checks and does both linting and style checks.
###ESLint
Please use ESLint with the eslint-config-vgno
-package from NPM:
npm install --save eslint-config-vgno
echo '{ "extends": "vgno" }' > .eslintrc
Alternatively, the .eslintrc
config found in this repository can be used.
###JSHint
Please use JSHint with the .jshintrc
config found in this
repository.
###JSCS
Please use JSCS with the .jscsrc
config found in this repository.